(*
 * Copyright (c) Facebook, Inc. and its affiliates.
 *
 * This source code is licensed under the MIT license found in the
 * LICENSE file in the root directory of this source tree.
 *)

module ArgSpec : sig
  type 'a flag_t

  type ('a, 'b) t

  type flag_arg_count =
    | No_Arg
    | Arg
    | Arg_List
    | Arg_Rest
    | Arg_Command

  type flag_metadata = {
    doc: string;
    env: string option;
    arg_count: flag_arg_count;
  }

  val empty : ('a, 'a) t

  val flag : string -> 'a flag_t -> doc:string -> ?env:string -> ('b, 'a -> 'c) t -> ('b, 'c) t

  val anon : string -> 'a flag_t -> ('b, 'a -> 'c) t -> ('b, 'c) t

  val rest : ('a, string list option -> 'b) t -> ('a, 'b) t

  val dummy : 'a -> ('b, 'a -> 'c) t -> ('b, 'c) t

  val collect : ('main -> 'a -> 'new_main) -> ('b, 'main) t -> ('b, 'a -> 'new_main) t

  val no_arg : bool flag_t

  val string : string option flag_t

  val bool : bool option flag_t

  val int : int option flag_t

  val uint : int option flag_t

  val enum : (string * 't) list -> 't option flag_t

  val command : (string * 'cmd) list -> ('cmd * string list) option flag_t

  val required : ?default:'a -> 'a option flag_t -> 'a flag_t

  val optional : 'a option flag_t -> 'a option flag_t

  val list_of : 'a option flag_t -> 'a list option flag_t

  val delimited : string -> 'a option flag_t -> 'a list option flag_t

  val key_value : string -> 'a option flag_t * 'b flag_t -> ('a * 'b) option flag_t
end

type ('a, 'b) builder_t = {
  name: string;
  doc: string;
  usage: string;
  args: ('a, 'b) ArgSpec.t;
}

type t

exception Show_help

exception Failed_to_parse of string * string

(* If no `col_width` is passed, the length of the longest string of the first column will be used.
   `col_pad` will be used as additional padding between the two columns. *)
val format_two_columns :
  ?margin:int -> ?col_width:int -> ?col_pad:int -> (string * string) list -> string

val usage : ('a, 'b) builder_t -> unit

val command : ('main, unit -> unit) builder_t -> 'main -> t

(* accessors *)
val run : t -> string list SMap.t -> unit

val name : t -> string

val doc : t -> string

val flags : t -> ArgSpec.flag_metadata SMap.t

val args_of_argv : t -> string list -> string list SMap.t

val string_of_usage : t -> string
